Page types

Page types (or master pages, page templates) are used to define margins for pages, create page frames and perform actions when a page is created or written to the PDF file. Classically, there is a page template for left pages and for right pages. In the simplest case, a template looks like this:

<Pagetype name="page" test="sd:even(sd:current-page())">
    <Margin left="1cm" right="1cm" top="1cm" bottom="1cm"/>
</Pagetype>

The condition in the attribute 'test' can be arbitrarily complex. This attribute must always be specified. A page is selected as soon as a test of a page type yields true. Examples:

  • true(): This page is always selected, since true() always results in true.
  • sd:current-page() > 1: Here the page template is selected for all pages following the first page.
  • sd:even(sd:current-page()): If the page number is even, this page type is used. This is usually when the page is a left page.
  • It can also be more complex: sd:even(sd:current-page()) and $part = 'main'. As long as the condition can be evaluated to true or false, this is a valid expression.

What happens if there are multiple conditions that are true at the same time? Page types are evaluated from “bottom to top”. This means that special page templates must be defined later than the general ones. The default template is defined first and has as condition true(). So it is evaluated last according to this logic and is always used if no other page type in the test returns true (fallback).

<Pagetype name="left pages" test="sd:even(sd:current-page())">
  ...
</Pagetype>
<Pagetype name="right pages" test="sd:odd(sd:current-page())">
  ...
</Pagetype>
<Pagetype name="first page" test="sd:current-page() = 1">
  ...
</Pagetype>

Here, the page type for the first page after the page type for right pages must be defined, otherwise it would not be considered (sd:odd(1) returns true).

Text frame

Text frames can be created in the page type definition. These are described in detail in the section "Areas on the page".

AtPageCreation, AtPageShipout

The two commands <AtPageCreation> and <AtPageShipout> are responsible for executing code when a page is created and when a page is written to the PDF file. They can be used for many purposes. The most common is to create the page header in <AtPageCreation> and the page footer in <AtPageShipout>.

The following is an example of a page footer.

<Layout xmlns="urn:speedata.de:2009/publisher/en"
  xmlns:sd="urn:speedata:2009/publisher/functions/en">

  <Pagetype name="page" test="true()">
    <Margin left="1cm" right="1cm" top="1cm" bottom="1cm"/>

    <AtPageShipout>
      <PlaceObject column="1" row="{sd:number-of-rows() - 1}">
        <Table stretch="max">
          <Tablerule/>
          <Tr>
            <Td align="left">
              <Paragraph>
                <Value select="sd:current-page()"/>
              </Paragraph>
            </Td>
            <Td align="right">
              <Paragraph>
                <Value>Name</Value>
              </Paragraph>
            </Td>
          </Tr>
        </Table>
      </PlaceObject>
    </AtPageShipout>
  </Pagetype>

  <Record element="data">
    <PlaceObject>
      <Textblock>
        <Paragraph>
          <Value>Content</Value>
        </Paragraph>
      </Textblock>
    </PlaceObject>
  </Record>
</Layout>
fusszeileseitentyp
Page with footer

ClearPage

With the command <ClearPage> you can specify which page type should be selected for the next page, even if the condition (test) for `<Pagetype>' does not return true.

The following example defines two page types, a template “Standard”, which is always used and a template “Special”, which is explicitly selected with <ClearPage>.

<Layout xmlns="urn:speedata.de:2009/publisher/en"
  xmlns:sd="urn:speedata:2009/publisher/functions/en">

  <Pageformat width="210mm" height="50mm"/>

  <Pagetype name="Special" test="false()">
    <Margin left="1cm" right="1cm" top="1cm" bottom="1cm"/>
  </Pagetype>

  <Pagetype name="Standard" test="true()">
    <Margin left="1cm" right="1cm" top="1cm" bottom="1cm"/>
  </Pagetype>

  <Record element="data">
    <PlaceObject>
      <Textblock>
        <Paragraph>
          <Value>Page 1</Value>
        </Paragraph>
      </Textblock>
    </PlaceObject>
    <ClearPage pagetype="Special" openon="right" />
    <PlaceObject>
      <Textblock>
        <Paragraph>
          <Value>Page 3</Value>
        </Paragraph>
      </Textblock>
    </PlaceObject>
  </Record>
</Layout>

In the log file (publisher-protocol.xml) you can see which page types are selected:

<entry level="INFO" msg="Create page" type="Standard" pagenumber="1"></entry>
<entry level="INFO" msg="Number of rows: 3, number of columns = 19"></entry>
<entry level="INFO" msg="Create font metrics" name="texgyreheros-regular.otf" size="10.0" id="1" mode="harfbuzz"></entry>
<entry level="INFO" msg="Shipout page 1"></entry>
<entry level="INFO" msg="Create page" type="Standard" pagenumber="2"></entry>
<entry level="INFO" msg="Number of rows: 3, number of columns = 19"></entry>
<entry level="INFO" msg="Shipout page 2"></entry>
<entry level="INFO" msg="Create page" type="Special" pagenumber="3"></entry>
<entry level="INFO" msg="Number of rows: 3, number of columns = 19"></entry>
<entry level="INFO" msg="Shipout page 3"></entry>

Pagenumbers

The page number of the current page can be determined with sd:current-page(). This function returns the number 1 on the first page, the number 2 on the second page and so on. The startpage attribute at <options> can be used to set the start page number:

<Layout xmlns="urn:speedata.de:2009/publisher/en"
    xmlns:sd="urn:speedata:2009/publisher/functions/en">
    <Options startpage="4"></Options>
    <Record element="data">
        <PlaceObject>
            <Textblock>
                <Paragraph>
                    <!-- prints out sd:current-page() = 4 -->
                    <Value>sd:current-page() = </Value>
                    <Value select="sd:current-page()" />
                </Paragraph>
            </Textblock>
        </PlaceObject>
    </Record>
</Layout>

Sections or matters

A document can be divided into sections (matters) that have their own numbering. The numbering of the sections does not change the numbering with sd:current-page(), rather they have “visible page numbers”. The visible page number can be determined with the function sd:visible-pagenumber(). A number must be passed to this function, otherwise it takes the current page. Without custom matters, the output is identical:

<!-- prints out sd:current-page() = 1 -->
<Value>sd:current-page() = </Value>
<Value select="sd:current-page()" />
<Br/>
<!-- sd:visible‐pagenumber(sd:current‐page()) = 1 -->
<Value>sd:visible-pagenumber(sd:current-page()) = </Value>
<Value select="sd:visible-pagenumber(sd:current-page())" />

If you define and use your own sections or use a predefined section, this can change:

<ClearPage matter="frontmatter" />
<PlaceObject>
    <Textblock>
        <Paragraph>
            <!-- prints out sd:current-page() = 1 -->
            <Value>sd:current-page() = </Value>
            <Value select="sd:current-page()" />
            <Br/>
            <!-- sd:visible‐pagenumber(sd:current‐page()) = i -->
            <Value>sd:visible-pagenumber(sd:current-page()) = </Value>
            <Value select="sd:visible-pagenumber(sd:current-page())" />
        </Paragraph>
    </Textblock>
</PlaceObject>

Here (with <Clearpage>) the matter frontmatter is used for the following pages. This causes the visible page number to be output as a Roman numeral in lower case, as this section has been defined as follows:

<DefineMatter name="frontmatter" label="lowercase-romannumeral" />

Matters can only start with the command <Clearpage>.